//==========================================
// Matt Pietrek
// Microsoft Systems Journal, Feb 1997
// FILE: EXEFILE.CPP
//==========================================
#include "stdafx.h"
//#pragma hdrstop
#include "exefile.h"

EXE_FILE::EXE_FILE(LPCTSTR pszFileName ) : MEMORY_MAPPED_FILE( pszFileName )
{
    m_errorType				= errEXE_FILE_FILE_NOT_FOUND;
    m_secondaryHeaderOffset = -1;   // A bogus value to catch bugs
    m_exeType				= exeType_Invalid;

    if ( FALSE == MEMORY_MAPPED_FILE::IsValid() )
        return;     // m_errorType already set to errEXE_FILE_FILE_NOT_FOUND

    // If we get here, the file exists, and was mapped.  We're still not
    // sure that it's a valid EXE though
    m_errorType = errEXE_FILE_INVALID_FORMAT;

    if ( GetFileSize() < sizeof(IMAGE_DOS_HEADER) )
        return;
    
    PIMAGE_DOS_HEADER pDosHdr = (PIMAGE_DOS_HEADER)GetBase();
    if ( IMAGE_DOS_SIGNATURE != pDosHdr->e_magic )
        return;

    // If we get here, it's at least a DOS 'MZ' file
    m_errorType = errEXE_FILE_NO_ERROR;

    if ( pDosHdr->e_lfarlc < 0x40 ) // Theoretically, this field must be >=
    {                               // 0x40 for it to be a non-DOS executable
        m_exeType = exeType_DOS;
        return;
    }

    // Sanity check.  Make sure the "new header" offset isn't past the end
    // of the file
    if ( pDosHdr->e_lfanew > (LONG)GetFileSize() )
        return;

    // Make a pointer to the secondary header   
    m_secondaryHeaderOffset = pDosHdr->e_lfanew;
    PWORD pSecondHdr = MakePtr( PWORD, GetBase(), m_secondaryHeaderOffset );

    // Decide what type of EXE, based on the start of the secondary header
    switch ( *pSecondHdr )
    {
        case IMAGE_OS2_SIGNATURE: m_exeType = exeType_NE; break;
        case IMAGE_VXD_SIGNATURE: m_exeType = exeType_VXD; break;
        case 0x4558: m_exeType = exeType_LX; break;     // OS/2 2.X
    }
    
    if ( *(PDWORD)pSecondHdr == IMAGE_NT_SIGNATURE )
        m_exeType = exeType_PE;
}

LPTSTR EXE_FILE::GetFileTypeDescription( void )
{
    // Returns a static string that describes what type this file is
    switch ( m_exeType )
    {
        case exeType_DOS:   return _T("DOS");
        case exeType_NE:    return _T("NE");
        case exeType_VXD:   return _T("VXD");
        case exeType_LX:    return _T("LX");
        case exeType_PE:    return _T("PE");
        default:            return _T("Invalid");
    }
}
